package com.myth.communication;

/**
 * 线程通信的例子:使用两个线程打印1-100。线程1，线程2交替打印
 *
 * 涉及到的三个方法:
 * wait():-旦执行此方法，当前线程就进入阻塞状态，并释放同步监视器。
 * notify(): -旦执行此方法，就会唤醒被wait的一一个线程。如果有多个线程被wait,就唤醒优先级高的那个线程。
 * notifyAll():一旦执行此方法，就会唤醒所有被wait的线程。
 *
 * 说明:
 * 1. wait(), notify(), notifyALl()三个方法必须使用在同步代码块或同步方法中。
 * 2. wait(), notify(), notifyAll()三个方法的调用者必须是同步代码块或同步方法中的同步监视器。
 *      否则，会出现IllegalMonitorStateException异常
 * 3. wait(), notify(), notifyALl()三个方法是定义在java. lang. object类中。
 *
 *
 * 面试题: sleep()和wait()的异同?
 * 1.相同点: -且执行方法，都可以使得当前的线程进入阻塞状态。
 * 2.不同点: 1)两个方法声明的位置不同: Thread类中声明sleep() ，object类中声明wait()
 *          2)调用的要求不同:sleep()可以在任何需要的场景下调用。 wait()必须使用在同步代码块或同步方法中
 *          3)关于是否释放同步监视器:如果两个方法都使用在同步代码块或同步方法中，sleep()不会释放锁，wait() 会释放锁。
 * @author
 * @date 2021-02-19
 */

public class CommunicationTest {
    public static void main(String[] args) {
        Number number = new Number();
        Thread t1 = new Thread(number);
        t1.setName("线程一");
        Thread t2 = new Thread(number);
        t2.setName("线程二");

        t1.start();
        t2.start();
    }
}

class Number implements Runnable {
    private int number = 1;

    @Override
    public void run() {
        while (true) {
            synchronized (this) {
                notify();
                if (number < 100) {
                    System.out.println(Thread.currentThread().getName() + ":" + number++);
                    try {
                        /**
                         * 使得调用如下方法的线程阻塞。一旦执行wait，会自动释放锁
                         */
                        wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                } else {
                    break;
                }
            }
        }
    }
}
